package com.ashbur.mybatis.binding;

import java.io.Serializable;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.util.Map;

/**
 * @author : Eumenides
 * @Description :
 * @date : 2022/7/15
 * @Copyright ： 公众号：菜鸟的大厂梦
 */
public class MapperProxy<T> implements InvocationHandler, Serializable {
    /**
     * 目前我们这里只是简单的封装了一个 sqlSession 的 Map 对象，你可以想象成所有的数据库语句操作，
     * 都是通过接口名称+方法名称作为key，操作作为逻辑的方式进行使用的。
     * 那么在反射调用中则获取对应的操作直接执行并返回结果即可。当然这还只是最核心的简化流程，后续不断补充内容后，会看到对数据库的操作
     */
    private Map<String, String> sqlSession;
    private final Class<T> mapperInterface;

    public MapperProxy(Map<String, String> sqlSession, Class<T> mapperInterface) {
        this.sqlSession = sqlSession;
        this.mapperInterface = mapperInterface;
    }

    /**
     * 通过实现 InvocationHandler#invoke 代理类接口，封装操作逻辑的方式，对外接口提供数据库操作对象
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        /**
         * 另外这里要注意如果是 Object 提供的 toString、hashCode 等方法是不需要代理执行的，
         * 所以添加 Object.class.equals(method.getDeclaringClass()) 判断。
         *
         * # getDeclaringClass：Method类的java.lang.reflect.Method.getDeclaringClass()方法返回该类或接口的Class对象
         */
        if (Object.class.equals(method.getDeclaringClass())) {
            return method.invoke(this, args);
        } else {
            return "你的被代理了！" + sqlSession.get(mapperInterface.getName() + "." + method.getName());
        }
    }
}
